home *** CD-ROM | disk | FTP | other *** search
- /* AX25 link callsign monitoring. Also contains beginnings of
- * an automatic link quality monitoring scheme (incomplete)
- *
- * Copyright 1991 Phil Karn, KA9Q
- */
- #include "global.h"
- #include "ctype.h"
- #include "mbuf.h"
- #include "iface.h"
- #include "pool.h"
-
- #if !defined(_lint)
- static char rcsid[] OPTIONAL = "$Id: axheard.c,v 1.13 1996/12/23 20:37:36 root Exp root $";
- #endif
-
- #define iscallsign(c) ((isupper(c)) || (isdigit(c)) || (c ==' '))
- int axheard_filter_flag = AXHEARD_PASS;
-
- static struct lq *al_create (struct iface * ifp, char *addr);
- static struct ld *ad_create (struct iface * ifp, const char *addr);
- struct lq *Lq;
- struct ld *Ld;
-
- extern int Maxax25heard;
-
- #ifdef POOLED
- struct mempool lq_pool = {NULLPOOLBLK, NULLPOOLBLK, 0, sizeof (struct lq), 50};
- struct mempool ld_pool = {NULLPOOLBLK, NULLPOOLBLK, 0, sizeof (struct ld), 50};
- #endif
-
-
- #if 0
- /* Send link quality reports to interface */
- void
- genrpt (ifp)
- struct iface *ifp;
- {
- struct mbuf *bp;
- register char *cp;
- int i;
- struct lq *lp;
- int maxentries, nentries;
-
- maxentries = (Paclen - LQHDR) / LQENTRY;
- if ((bp = alloc_mbuf (Paclen)) == NULLBUF)
- return;
- cp = bp->data;
- nentries = 0;
-
- /* Build and emit header */
- cp = putlqhdr (cp, LINKVERS, Ip_addr);
-
- /* First entry is for ourselves. Since we're examining the Axsent
- * variable before we've sent this frame, add one to it so it'll
- * match the receiver's count after he gets this frame.
- */
- cp = putlqentry (cp, ifp->hwaddr, Axsent + 1);
- nentries++;
-
- /* Now add entries from table */
- for (lp = lq; lp != NULLLQ; lp = lp->next) {
- cp = putlqentry (cp, &lp->addr, lp->currxcnt);
- if (++nentries >= MAXENTRIES) {
- /* Flush */
- bp->cnt = nentries * LQENTRY + LQHDR;
- ax_output (ifp, Ax25multi[0], ifp->hwaddr, PID_LQ, bp);
- if ((bp = alloc_mbuf (Paclen)) == NULLBUF)
- return;
- cp = bp->data;
- }
- }
- if (nentries > 0) {
- bp->cnt = nentries * LQENTRY + LQHDR;
- ax_output (ifp, Ax25multi[0], ifp->hwaddr, LQPID, bp);
- } else
- free_p (bp);
- }
-
-
- /* Pull the header off a link quality packet */
- void
- getlqhdr (hp, bpp)
- struct lqhdr *hp;
- struct mbuf **bpp;
- {
- hp->version = pull16 (bpp);
- hp->ip_addr = pull32 (bpp);
- }
-
-
- /* Put a header on a link quality packet.
- * Return pointer to buffer immediately following header
- */
- char *
- putlqhdr (cp, version, ip_addr)
- register char *cp;
- int16 version;
- uint32 ip_addr;
- {
- cp = put16 (cp, version);
- return put32 (cp, ip_addr);
- }
-
-
- /* Pull an entry off a link quality packet */
- void
- getlqentry (ep, bpp)
- struct lqentry *ep;
- struct mbuf **bpp;
- {
- pullup (bpp, ep->addr, AXALEN);
- ep->count = pull32 (bpp);
- }
-
-
- /* Put an entry on a link quality packet
- * Return pointer to buffer immediately following header
- */
- char *
- putlqentry (cp, addr, count)
- char *cp;
- char *addr;
- int32 count;
- {
- memcpy (cp, addr, AXALEN);
- cp += AXALEN;
- return put32 (cp, count);
- }
- #endif
-
-
- /* Log the source address of an incoming packet */
- void
- logsrc (ifp, addr)
- struct iface *ifp;
- char *addr;
- {
- register struct lq *lp;
- register unsigned char c;
- register int i = 0;
-
- if (axheard_filter_flag & AXHEARD_NOSRC || !(ifp->flags & LOG_AXHEARD))
- return;
-
- while (i < AXALEN - 1) {
- c = uchar(*(addr + i));
- c >>= 1;
- if (!iscallsign (c))
- return;
- i++;
- }
-
- if ((lp = al_lookup (ifp, addr, 1)) == NULLLQ)
- if ((lp = al_create (ifp, addr)) == NULLLQ)
- return;
- lp->currxcnt++;
- lp->time = secclock ();
- }
-
-
- /* Log the destination address of an incoming packet */
- void
- logdest (ifp, addr)
- struct iface *ifp;
- const char *addr;
- {
- register struct ld *lp;
- register unsigned char c;
- register int i = 0;
-
- if (axheard_filter_flag & AXHEARD_NODST || !(ifp->flags & LOG_AXHEARD))
- return;
-
- while (i < AXALEN - 1) {
- c = uchar(*(addr + i));
- c >>= 1;
- if (!iscallsign (c))
- return;
- i++;
- }
-
- if ((lp = ad_lookup (ifp, addr, 1)) == NULLLD)
- if ((lp = ad_create (ifp, addr)) == NULLLD)
- return;
- lp->currxcnt++;
- lp->time = secclock ();
- }
-
-
- /* Look up an entry in the source data base */
- struct lq *
- al_lookup (ifp, addr, sort)
- struct iface *ifp;
- char *addr;
- int sort;
- {
- register struct lq *lp;
- struct lq *lplast = NULLLQ;
-
- for (lp = Lq; lp != NULLLQ; lplast = lp, lp = lp->next) {
- if (addreq (lp->addr, addr) && (lp->iface == ifp)) {
- if (sort && lplast != NULLLQ) {
- /* Move entry to top of list */
- lplast->next = lp->next;
- lp->next = Lq;
- Lq = lp;
- }
- return lp;
- }
- }
- return NULLLQ;
- }
-
-
- /* Create a new entry in the source database */
- /* If there are too many entries, override the oldest one - WG7J */
- static struct lq *
- al_create (ifp, addr)
- struct iface *ifp;
- char *addr;
- {
- static int numdb = 0;
- register struct lq *lp;
- struct lq *lplast = NULLLQ;
-
- if (Maxax25heard && numdb == Maxax25heard) {
- /* find and use last one in list */
- for (lp = Lq; lp->next != NULLLQ; lplast = lp, lp = lp->next)
- ;
- /* delete entry from end */
- if (lplast)
- lplast->next = NULLLQ;
- } else { /* create a new entry */
- numdb++;
- #ifdef POOLED
- lp = (struct lq *) pool_alloc (&lq_pool);
- #else
- lp = (struct lq *) callocw (1, sizeof (struct lq));
- #endif
- }
- memcpy (lp->addr, addr, AXALEN);
- lp->iface = ifp;
- lp->next = Lq;
- Lq = lp;
-
- return lp;
- }
-
-
- /* Look up an entry in the destination database */
- struct ld *
- ad_lookup (ifp, addr, sort)
- struct iface *ifp;
- const char *addr;
- int sort;
- {
- register struct ld *lp;
- struct ld *lplast = NULLLD;
-
- for (lp = Ld; lp != NULLLD; lplast = lp, lp = lp->next) {
- if ((lp->iface == ifp) && addreq (lp->addr, addr)) {
- if (sort && lplast != NULLLD) {
- /* Move entry to top of list */
- lplast->next = lp->next;
- lp->next = Ld;
- Ld = lp;
- }
- return lp;
- }
- }
- return NULLLD;
- }
-
-
- /* Create a new entry in the destination database */
- static struct ld *
- ad_create (ifp, addr)
- struct iface *ifp;
- const char *addr;
- {
- static int numdb = 0;
- register struct ld *lp;
- struct ld *lplast = NULLLD;
-
- if (Maxax25heard && numdb == Maxax25heard) { /* find and use last one in list */
- for (lp = Ld; lp->next != NULLLD; lplast = lp, lp = lp->next)
- ;
- /* delete entry from end */
- if (lplast)
- lplast->next = NULLLD;
- } else { /* create a new entry */
- numdb++;
- #ifdef POOLED
- lp = (struct ld *) pool_alloc (&ld_pool);
- #else
- lp = (struct ld *) callocw (1, sizeof (struct ld));
- #endif
- }
- memcpy (lp->addr, addr, AXALEN);
- lp->iface = ifp;
- lp->next = Ld;
- Ld = lp;
-
- return lp;
- }
-